1 |
|
2 |
|
3 |
|
4 |
|
5 |
|
6 | import fs from 'fs'
|
7 | import path from 'path'
|
8 | import Module from 'module'
|
9 |
|
10 | import Log from './tools/log'
|
11 | import { ends_with } from './helpers'
|
12 |
|
13 | const original_findPath = Module._findPath
|
14 |
|
15 | export default class Require_hacker
|
16 | {
|
17 | preceding_abstract_path_resolvers = []
|
18 | abstract_path_resolvers = []
|
19 |
|
20 |
|
21 |
|
22 | abstract_path_resolved_modules = {}
|
23 |
|
24 | constructor(options)
|
25 | {
|
26 |
|
27 |
|
28 |
|
29 |
|
30 | this.log = new Log('require-hook', { debug: options.debug })
|
31 |
|
32 |
|
33 |
|
34 | Module._findPath = (...parameters) =>
|
35 | {
|
36 | const request = parameters[0]
|
37 |
|
38 |
|
39 |
|
40 | for (let resolver of this.preceding_abstract_path_resolvers)
|
41 | {
|
42 | const resolved = resolver.resolve(request)
|
43 | if (typeof resolved !== 'undefined')
|
44 | {
|
45 | return resolved
|
46 | }
|
47 | }
|
48 |
|
49 |
|
50 | const filename = original_findPath.apply(undefined, parameters)
|
51 | if (filename !== false)
|
52 | {
|
53 | return filename
|
54 | }
|
55 |
|
56 |
|
57 | for (let resolver of this.abstract_path_resolvers)
|
58 | {
|
59 | const resolved = resolver.resolve(request)
|
60 | if (typeof resolved !== 'undefined')
|
61 | {
|
62 | return resolved
|
63 | }
|
64 | }
|
65 |
|
66 | return false
|
67 | }
|
68 | }
|
69 |
|
70 |
|
71 |
|
72 |
|
73 |
|
74 |
|
75 |
|
76 |
|
77 |
|
78 |
|
79 |
|
80 |
|
81 |
|
82 |
|
83 |
|
84 |
|
85 |
|
86 |
|
87 |
|
88 |
|
89 |
|
90 | resolver(id, resolver, options = {})
|
91 | {
|
92 | validate.resolver(id, resolver)
|
93 |
|
94 | const resolver_entry =
|
95 | {
|
96 | id,
|
97 | resolve: path =>
|
98 | {
|
99 | const resolved_path = `${path}.${id}`
|
100 |
|
101 |
|
102 | const source = resolver(path)
|
103 |
|
104 | if (typeof source === 'undefined')
|
105 | {
|
106 | return
|
107 | }
|
108 |
|
109 |
|
110 | delete require.cache[resolved_path]
|
111 |
|
112 | this.abstract_path_resolved_modules[resolved_path] = source
|
113 |
|
114 | return resolved_path
|
115 | }
|
116 | }
|
117 |
|
118 | if (options.precede_node_loader)
|
119 | {
|
120 | this.preceding_abstract_path_resolvers.push(resolver_entry)
|
121 | }
|
122 | else
|
123 | {
|
124 | this.abstract_path_resolvers.push(resolver_entry)
|
125 | }
|
126 |
|
127 | const hook = this.hook(id, path =>
|
128 | {
|
129 | const source = this.abstract_path_resolved_modules[path]
|
130 | delete this.abstract_path_resolved_modules[path]
|
131 | return source
|
132 | })
|
133 |
|
134 | const result =
|
135 | {
|
136 | unmount: () =>
|
137 | {
|
138 |
|
139 | this.preceding_abstract_path_resolvers = this.preceding_abstract_path_resolvers.filter(x => x !== resolver_entry)
|
140 | this.abstract_path_resolvers = this.abstract_path_resolvers.filter(x => x !== resolver_entry)
|
141 | hook.unmount()
|
142 | }
|
143 | }
|
144 |
|
145 | return result
|
146 | }
|
147 |
|
148 |
|
149 |
|
150 |
|
151 |
|
152 |
|
153 |
|
154 |
|
155 |
|
156 |
|
157 |
|
158 |
|
159 |
|
160 |
|
161 |
|
162 |
|
163 | hook(extension, resolve)
|
164 | {
|
165 | this.log.debug(`Hooking into *.${extension} files loading`)
|
166 |
|
167 |
|
168 | validate.extension(extension)
|
169 | validate.resolve(resolve)
|
170 |
|
171 |
|
172 | const dot_extension = `.${extension}`
|
173 |
|
174 |
|
175 | const original_loader = Module._extensions[dot_extension]
|
176 |
|
177 |
|
178 | if (original_loader)
|
179 | {
|
180 | const output = (extension === 'js' ? this.log.debug : this.log.warning).bind(this.log)
|
181 | output(`-----------------------------------------------`)
|
182 | output(`Overriding an already existing require() hook `)
|
183 | output(`for file extension ${dot_extension}`)
|
184 | output(`-----------------------------------------------`)
|
185 | }
|
186 |
|
187 |
|
188 | const cached_modules = new Set()
|
189 |
|
190 |
|
191 | Module._extensions[dot_extension] = (module, filename) =>
|
192 | {
|
193 | this.log.debug(`Loading source code for ${filename}`)
|
194 |
|
195 |
|
196 | let aborted = false
|
197 |
|
198 |
|
199 | const source = resolve(filename, () =>
|
200 | {
|
201 | this.log.debug(`Fallback to original loader`)
|
202 |
|
203 |
|
204 | aborted = true
|
205 |
|
206 |
|
207 |
|
208 | if (path.extname(filename) !== dot_extension)
|
209 | {
|
210 | this.log.info(`Trying to load "${path.basename(filename)}" as a "*${dot_extension}"`)
|
211 | }
|
212 |
|
213 |
|
214 | (original_loader || Module._extensions['.js'])(module, filename)
|
215 | })
|
216 |
|
217 |
|
218 | if (aborted)
|
219 | {
|
220 | return
|
221 | }
|
222 |
|
223 |
|
224 | cached_modules.add(filename)
|
225 |
|
226 |
|
227 |
|
228 | module._compile(source, filename)
|
229 | }
|
230 |
|
231 | const result =
|
232 | {
|
233 |
|
234 | unmount: () =>
|
235 | {
|
236 |
|
237 | for (let path of cached_modules)
|
238 | {
|
239 | delete require.cache[path]
|
240 | }
|
241 |
|
242 |
|
243 | Module._extensions[dot_extension] = original_loader
|
244 | }
|
245 | }
|
246 |
|
247 | return result
|
248 | }
|
249 |
|
250 |
|
251 |
|
252 |
|
253 |
|
254 |
|
255 |
|
256 |
|
257 |
|
258 |
|
259 |
|
260 |
|
261 |
|
262 |
|
263 |
|
264 |
|
265 |
|
266 |
|
267 |
|
268 |
|
269 |
|
270 |
|
271 |
|
272 |
|
273 |
|
274 |
|
275 | }
|
276 |
|
277 |
|
278 | const validate =
|
279 | {
|
280 | extension(extension)
|
281 | {
|
282 | if (typeof extension !== 'string')
|
283 | {
|
284 | throw new Error(`Expected string extension. Got ${extension}`)
|
285 | }
|
286 |
|
287 | if (path.extname(`test.${extension}`) !== `.${extension}`)
|
288 | {
|
289 | throw new Error(`Invalid file extension ${extension}`)
|
290 | }
|
291 | },
|
292 |
|
293 | resolve(resolve)
|
294 | {
|
295 | if (typeof resolve !== 'function')
|
296 | {
|
297 | throw new Error(`Resolve should be a function. Got ${resolve}`)
|
298 | }
|
299 | },
|
300 |
|
301 | resolver(id, resolver)
|
302 | {
|
303 | if (!id)
|
304 | {
|
305 | throw new Error(`You must specify resolver id`)
|
306 | }
|
307 |
|
308 | if (path.extname(`test.${id}`) !== `.${id}`)
|
309 | {
|
310 | throw new Error(`Invalid resolver id. Expected a valid file extension.`)
|
311 | }
|
312 | }
|
313 | } |
\ | No newline at end of file |